Line | Count | Source |
1 | | // Copyright (c) 2024, BlockProject 3D |
2 | | // |
3 | | // All rights reserved. |
4 | | // |
5 | | // Redistribution and use in source and binary forms, with or without modification, |
6 | | // are permitted provided that the following conditions are met: |
7 | | // |
8 | | // * Redistributions of source code must retain the above copyright notice, |
9 | | // this list of conditions and the following disclaimer. |
10 | | // * Redistributions in binary form must reproduce the above copyright notice, |
11 | | // this list of conditions and the following disclaimer in the documentation |
12 | | // and/or other materials provided with the distribution. |
13 | | // * Neither the name of BlockProject 3D nor the names of its contributors |
14 | | // may be used to endorse or promote products derived from this software |
15 | | // without specific prior written permission. |
16 | | // |
17 | | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
18 | | // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
19 | | // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
20 | | // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR |
21 | | // CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
22 | | // EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
23 | | // PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
24 | | // PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
25 | | // LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
26 | | // NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
27 | | // SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
28 | | |
29 | | use crate::compiler::error::Error; |
30 | | use crate::compiler::imports::Import; |
31 | | use crate::compiler::message::{FieldType, Message}; |
32 | | use crate::compiler::r#enum::Enum; |
33 | | use crate::compiler::structure::Structure; |
34 | | use crate::compiler::union::Union; |
35 | | use crate::compiler::util::imports::ImportSolver; |
36 | | use crate::compiler::util::objects::{name_index, ObjectStore}; |
37 | | use crate::compiler::util::types::{Name, TypePathMap}; |
38 | | use crate::model::message::MessageFieldValue; |
39 | | use crate::model::protocol::{Description, Endianness}; |
40 | | use crate::model::typedef::Typedef; |
41 | | use bp3d_debug::{info, trace}; |
42 | | use std::borrow::Cow; |
43 | | use std::collections::BTreeMap; |
44 | | use std::rc::Rc; |
45 | | use crate::compiler::util::protocols::ProtocolStore; |
46 | | |
47 | | name_index!(Typedef => name); |
48 | | |
49 | | #[derive(Clone, Debug)] |
50 | | pub struct Protocol { |
51 | | pub full_name: String, |
52 | | pub description: Option<Description>, |
53 | | pub endianness: Endianness, |
54 | | pub type_path_map: TypePathMap, |
55 | | pub structs: ObjectStore<Structure>, |
56 | | pub messages: ObjectStore<Message>, |
57 | | pub enums: ObjectStore<Enum>, |
58 | | pub unions: ObjectStore<Union>, |
59 | | pub types: ObjectStore<Typedef>, |
60 | | } |
61 | | |
62 | | impl bp3d_util::index_map::Index for Protocol { |
63 | | type Key = str; |
64 | | |
65 | 178 | fn index(&self) -> &Self::Key { |
66 | 178 | &self.full_name |
67 | 178 | } |
68 | | } |
69 | | |
70 | | impl Protocol { |
71 | 37 | pub fn name(&self) -> &str { |
72 | 37 | if let Some(id1 ) = self.full_name.rfind("::") { Branch (72:16): [True: 1, False: 36]
Branch (72:16): [Folded - Ignored]
|
73 | 1 | &self.full_name[id + 2..] |
74 | | } else { |
75 | 36 | &self.full_name |
76 | | } |
77 | 37 | } |
78 | | |
79 | 39 | pub fn package(&self) -> &str { |
80 | 39 | if let Some(id1 ) = self.full_name.rfind("::") { Branch (80:16): [True: 1, False: 38]
Branch (80:16): [Folded - Ignored]
|
81 | 1 | &self.full_name[..id] |
82 | | } else { |
83 | 38 | "" |
84 | | } |
85 | 39 | } |
86 | | |
87 | 76 | pub fn iter_codecs(&self) -> impl Iterator<Item = &str> { |
88 | 126 | self.messages.iter().flat_map(|v| v.fields.iter().filter_map(66 |v| v.codec.as_deref())66 ) |
89 | 76 | } |
90 | | |
91 | 58 | pub fn from_model<T: ImportSolver, U>( |
92 | 58 | mut value: crate::model::Protocol, |
93 | 58 | protocols: &ProtocolStore<T, U>, |
94 | 58 | package: &str, |
95 | 58 | ) -> Result<Self, Error> { |
96 | 58 | let full_name = if package.is_empty() { Branch (96:28): [True: 18, False: 2]
Branch (96:28): [Folded - Ignored]
Branch (96:28): [True: 18, False: 0]
Branch (96:28): [Folded - Ignored]
Branch (96:28): [True: 5, False: 0]
Branch (96:28): [True: 1, False: 0]
Branch (96:28): [True: 1, False: 0]
Branch (96:28): [True: 4, False: 0]
Branch (96:28): [True: 3, False: 0]
Branch (96:28): [True: 5, False: 0]
Branch (96:28): [True: 1, False: 0]
|
97 | 56 | value.name |
98 | | } else { |
99 | 2 | format!("{}::{}", package, value.name) |
100 | | }; |
101 | 58 | trace!("Compiling protocol {}", full_name); |
102 | 58 | let mut proto = Protocol { |
103 | 58 | full_name, |
104 | 58 | description: value.description, |
105 | 58 | endianness: value.endianness.unwrap_or(Endianness::Little), |
106 | 58 | type_path_map: TypePathMap::new(), |
107 | 58 | structs: ObjectStore::new(), |
108 | 58 | messages: ObjectStore::new(), |
109 | 58 | enums: ObjectStore::new(), |
110 | 58 | unions: ObjectStore::new(), |
111 | 58 | types: ObjectStore::new(), |
112 | 58 | }; |
113 | 58 | info!("Running import solver pass..."); |
114 | 58 | if let Some(mut imports11 ) = value.imports { Branch (114:16): [True: 6, False: 14]
Branch (114:16): [Folded - Ignored]
Branch (114:16): [True: 5, False: 13]
Branch (114:16): [Folded - Ignored]
Branch (114:16): [True: 0, False: 5]
Branch (114:16): [True: 0, False: 1]
Branch (114:16): [True: 0, False: 1]
Branch (114:16): [True: 0, False: 4]
Branch (114:16): [True: 0, False: 3]
Branch (114:16): [True: 0, False: 5]
Branch (114:16): [True: 0, False: 1]
|
115 | 11 | let mut solved_imports = Vec::new(); |
116 | 50 | while let Some(v39 ) = imports.pop() { Branch (116:23): [True: 20, False: 6]
Branch (116:23): [Folded - Ignored]
Branch (116:23): [True: 19, False: 5]
Branch (116:23): [Folded - Ignored]
Branch (116:23): [True: 0, False: 0]
Branch (116:23): [True: 0, False: 0]
Branch (116:23): [True: 0, False: 0]
Branch (116:23): [True: 0, False: 0]
Branch (116:23): [True: 0, False: 0]
Branch (116:23): [True: 0, False: 0]
Branch (116:23): [True: 0, False: 0]
|
117 | 39 | let protocol_path = if package.is_empty() || v.protocol.contains("::")1 { Branch (117:40): [True: 19, False: 1]
Branch (117:62): [True: 1, False: 0]
Branch (117:40): [Folded - Ignored]
Branch (117:62): [Folded - Ignored]
Branch (117:40): [True: 19, False: 0]
Branch (117:62): [True: 0, False: 0]
Branch (117:40): [Folded - Ignored]
Branch (117:62): [Folded - Ignored]
Branch (117:40): [True: 0, False: 0]
Branch (117:62): [True: 0, False: 0]
Branch (117:40): [True: 0, False: 0]
Branch (117:62): [True: 0, False: 0]
Branch (117:40): [True: 0, False: 0]
Branch (117:62): [True: 0, False: 0]
Branch (117:40): [True: 0, False: 0]
Branch (117:62): [True: 0, False: 0]
Branch (117:40): [True: 0, False: 0]
Branch (117:62): [True: 0, False: 0]
Branch (117:40): [True: 0, False: 0]
Branch (117:62): [True: 0, False: 0]
Branch (117:40): [True: 0, False: 0]
Branch (117:62): [True: 0, False: 0]
|
118 | 39 | Cow::Borrowed(&v.protocol) |
119 | | } else { |
120 | 0 | Cow::Owned(format!("{}::{}", package, v.protocol)) |
121 | | }; |
122 | 39 | trace!({protocol=&**protocol_path} {type=&*v.type_name}, "Solving import"); |
123 | 39 | let r = protocols.get(&protocol_path); |
124 | 39 | let r = match r { |
125 | 39 | Some(r) => r, |
126 | 0 | None => return Err(Error::UndefinedReference(v.protocol)), |
127 | | }; |
128 | 39 | let ty = r |
129 | 39 | .structs |
130 | 39 | .get(&v.type_name) |
131 | 39 | .map(Import::Struct) |
132 | 39 | .or_else(|| r.messages.get(&v.type_name).map(Import::Message)13 ) |
133 | 39 | .or_else(|| r.unions.get(&v.type_name).map(Import::Union)6 ) |
134 | 39 | .or_else(|| r.enums.get(&v.type_name).map(Import::Enum)4 ) |
135 | 39 | .or_else(|| r.types.get(&v.type_name).map(Import::Type)2 ) |
136 | 39 | .ok_or(Error::UnresolvedImport(format!("{}::{}", protocol_path, v.type_name)))?0 ; |
137 | 39 | let type_path = protocols.get_full_type_path(r, &v.type_name).ok_or(Error::SolverError)?0 ; |
138 | 39 | solved_imports.push(ty); |
139 | 160 | let count1 = imports.iter().filter(|vv| vv.type_name == v.type_name).count(); |
140 | 199 | let count2 = solved_imports.iter().filter(|vv| vv.name() == v.type_name).count()39 ; |
141 | 39 | let is_ambiguous = count1 > 0 || count2 > 137 ; Branch (141:36): [True: 1, False: 19]
Branch (141:36): [Folded - Ignored]
Branch (141:36): [True: 1, False: 18]
Branch (141:36): [Folded - Ignored]
Branch (141:36): [True: 0, False: 0]
Branch (141:36): [True: 0, False: 0]
Branch (141:36): [True: 0, False: 0]
Branch (141:36): [True: 0, False: 0]
Branch (141:36): [True: 0, False: 0]
Branch (141:36): [True: 0, False: 0]
Branch (141:36): [True: 0, False: 0]
|
142 | 39 | proto.type_path_map.add(&ty, type_path); |
143 | 39 | if is_ambiguous { Branch (143:20): [True: 2, False: 18]
Branch (143:20): [Folded - Ignored]
Branch (143:20): [True: 2, False: 17]
Branch (143:20): [Folded - Ignored]
Branch (143:20): [True: 0, False: 0]
Branch (143:20): [True: 0, False: 0]
Branch (143:20): [True: 0, False: 0]
Branch (143:20): [True: 0, False: 0]
Branch (143:20): [True: 0, False: 0]
Branch (143:20): [True: 0, False: 0]
Branch (143:20): [True: 0, False: 0]
|
144 | 4 | trace!({protocol=&**protocol_path} {type=&*v.type_name}, "Import is ambiguous, import it as {}::{}", v.protocol, v.type_name); |
145 | 4 | ty.insert(format!("{}::{}", v.protocol, v.type_name), &mut proto); |
146 | 35 | } else { |
147 | 35 | trace!({protocol=&**protocol_path} {type=&*v.type_name}, "Import is not ambiguous"); |
148 | 35 | ty.insert(v.type_name, &mut proto); |
149 | 35 | } |
150 | | } |
151 | 47 | } |
152 | 58 | info!("Adding typedefs..."); |
153 | 58 | if let Some(types4 ) = value.types { Branch (153:16): [True: 2, False: 18]
Branch (153:16): [Folded - Ignored]
Branch (153:16): [True: 2, False: 16]
Branch (153:16): [Folded - Ignored]
Branch (153:16): [True: 0, False: 5]
Branch (153:16): [True: 0, False: 1]
Branch (153:16): [True: 0, False: 1]
Branch (153:16): [True: 0, False: 4]
Branch (153:16): [True: 0, False: 3]
Branch (153:16): [True: 0, False: 5]
Branch (153:16): [True: 0, False: 1]
|
154 | 16 | for v12 in types { |
155 | 12 | trace!({model=?&v}, "Adding typedef to protocol"); |
156 | 12 | proto.types.insert(Rc::new(v)); |
157 | 12 | } |
158 | 54 | } |
159 | 58 | info!("Running type inference pass..."); |
160 | 58 | if let Some(structs47 ) = &mut value.structs { Branch (160:16): [True: 17, False: 3]
Branch (160:16): [Folded - Ignored]
Branch (160:16): [True: 16, False: 2]
Branch (160:16): [Folded - Ignored]
Branch (160:16): [True: 0, False: 5]
Branch (160:16): [True: 1, False: 0]
Branch (160:16): [True: 1, False: 0]
Branch (160:16): [True: 4, False: 0]
Branch (160:16): [True: 3, False: 0]
Branch (160:16): [True: 5, False: 0]
Branch (160:16): [True: 0, False: 1]
|
161 | 123 | for v76 in structs { |
162 | 209 | for field133 in &mut v.fields { |
163 | 0 | if let Some(info) = Branch (163:28): [True: 0, False: 60]
Branch (163:28): [Folded - Ignored]
Branch (163:28): [True: 0, False: 59]
Branch (163:28): [Folded - Ignored]
Branch (163:28): [True: 0, False: 0]
Branch (163:28): [True: 0, False: 1]
Branch (163:28): [True: 0, False: 1]
Branch (163:28): [True: 0, False: 4]
Branch (163:28): [True: 0, False: 2]
Branch (163:28): [True: 0, False: 6]
Branch (163:28): [True: 0, False: 0]
|
164 | 133 | field.item_type.as_ref().and_then(|v| proto.types.get(v)13 ).and_then(|v| v.to_struct()0 ) |
165 | 0 | { |
166 | 0 | trace!({typedef=?info}, "Inferred {} as {}", field.name, info.name); |
167 | 0 | let name = std::mem::replace(field, info.clone()).name; |
168 | 0 | field.name = name; |
169 | 133 | } |
170 | | } |
171 | | } |
172 | 11 | } |
173 | 58 | if let Some(messages28 ) = &mut value.messages { Branch (173:16): [True: 10, False: 10]
Branch (173:16): [Folded - Ignored]
Branch (173:16): [True: 8, False: 10]
Branch (173:16): [Folded - Ignored]
Branch (173:16): [True: 5, False: 0]
Branch (173:16): [True: 0, False: 1]
Branch (173:16): [True: 0, False: 1]
Branch (173:16): [True: 0, False: 4]
Branch (173:16): [True: 0, False: 3]
Branch (173:16): [True: 5, False: 0]
Branch (173:16): [True: 0, False: 1]
|
174 | 72 | for v44 in messages { |
175 | 122 | for field78 in &mut v.fields { |
176 | 12 | if let Some(info) = Branch (176:28): [True: 6, False: 30]
Branch (176:28): [Folded - Ignored]
Branch (176:28): [True: 6, False: 21]
Branch (176:28): [Folded - Ignored]
Branch (176:28): [True: 0, False: 6]
Branch (176:28): [True: 0, False: 0]
Branch (176:28): [True: 0, False: 0]
Branch (176:28): [True: 0, False: 0]
Branch (176:28): [True: 0, False: 0]
Branch (176:28): [True: 0, False: 9]
Branch (176:28): [True: 0, False: 0]
|
177 | 78 | field.item_type.as_ref().and_then(|v| proto.types.get(v)32 ).and_then(|v| v.to_message()12 ) |
178 | 12 | { |
179 | 12 | trace!({typedef=?info}, "Inferred {} as {}", field.name, info.name); |
180 | 12 | let name = std::mem::replace(field, info.clone()).name; |
181 | 12 | field.name = name; |
182 | 66 | } |
183 | | } |
184 | | } |
185 | 30 | } |
186 | 58 | info!("Running compiler pass..."); |
187 | 58 | if let Some(enums7 ) = value.enums { Branch (187:16): [True: 3, False: 17]
Branch (187:16): [Folded - Ignored]
Branch (187:16): [True: 3, False: 15]
Branch (187:16): [Folded - Ignored]
Branch (187:16): [True: 0, False: 5]
Branch (187:16): [True: 0, False: 1]
Branch (187:16): [True: 0, False: 1]
Branch (187:16): [True: 0, False: 4]
Branch (187:16): [True: 0, False: 3]
Branch (187:16): [True: 0, False: 5]
Branch (187:16): [True: 1, False: 0]
|
188 | 7 | trace!(">> Compiling enums..."); |
189 | 13 | for v7 in enums { |
190 | 7 | trace!({model=?&v}, "Compiling enum"); |
191 | 7 | let v6 = Rc::new(Enum::from_model(v)?1 ); |
192 | 6 | proto.enums.insert(v); |
193 | | } |
194 | 51 | } |
195 | 57 | if let Some(structs47 ) = value.structs { Branch (195:16): [True: 17, False: 3]
Branch (195:16): [Folded - Ignored]
Branch (195:16): [True: 16, False: 2]
Branch (195:16): [Folded - Ignored]
Branch (195:16): [True: 0, False: 5]
Branch (195:16): [True: 1, False: 0]
Branch (195:16): [True: 1, False: 0]
Branch (195:16): [True: 4, False: 0]
Branch (195:16): [True: 3, False: 0]
Branch (195:16): [True: 5, False: 0]
Branch (195:16): [True: 0, False: 0]
|
196 | 47 | trace!(">> Compiling structures..."); |
197 | 114 | for v76 in structs { |
198 | 76 | trace!({model=?&v}, "Compiling structure"); |
199 | 76 | let v67 = Rc::new(Structure::from_model(&proto, v)?9 ); |
200 | 67 | proto.structs.insert(v); |
201 | | } |
202 | 10 | } |
203 | 48 | let mut union_messages = BTreeMap::new(); |
204 | 48 | if let Some(mut messages28 ) = value.messages { Branch (204:16): [True: 10, False: 10]
Branch (204:16): [Folded - Ignored]
Branch (204:16): [True: 8, False: 10]
Branch (204:16): [Folded - Ignored]
Branch (204:16): [True: 5, False: 0]
Branch (204:16): [True: 0, False: 0]
Branch (204:16): [True: 0, False: 0]
Branch (204:16): [True: 0, False: 0]
Branch (204:16): [True: 0, False: 0]
Branch (204:16): [True: 5, False: 0]
Branch (204:16): [True: 0, False: 0]
|
205 | 28 | trace!(">> Extracting unions and messages with references to unions..."); |
206 | 28 | let len = messages.len(); |
207 | 44 | for i in 1..len + 128 { |
208 | 44 | let i = len - i; |
209 | 78 | let has_unions = messages[i].fields.iter().any(|v| match &v.value { |
210 | 20 | None => false, |
211 | 58 | Some(v) => matches!49 (v, MessageFieldValue::Union { .. }), |
212 | 78 | }); |
213 | 44 | if has_unions { Branch (213:20): [True: 2, False: 16]
Branch (213:20): [Folded - Ignored]
Branch (213:20): [True: 2, False: 13]
Branch (213:20): [Folded - Ignored]
Branch (213:20): [True: 0, False: 6]
Branch (213:20): [True: 0, False: 0]
Branch (213:20): [True: 0, False: 0]
Branch (213:20): [True: 0, False: 0]
Branch (213:20): [True: 0, False: 0]
Branch (213:20): [True: 5, False: 0]
Branch (213:20): [True: 0, False: 0]
|
214 | 9 | let msg = messages.remove(i); |
215 | 9 | union_messages.insert(i, msg); |
216 | 35 | } |
217 | | } |
218 | 28 | let pos = len; |
219 | 28 | let len = messages.len(); |
220 | 35 | for i in 1..len + 128 { |
221 | 35 | let i = len - i; |
222 | 35 | let has_ref = union_messages.values().any(|union| messages[i].references(&union.name)6 ); |
223 | 35 | if has_ref { Branch (223:20): [True: 1, False: 15]
Branch (223:20): [Folded - Ignored]
Branch (223:20): [True: 1, False: 12]
Branch (223:20): [Folded - Ignored]
Branch (223:20): [True: 0, False: 6]
Branch (223:20): [True: 0, False: 0]
Branch (223:20): [True: 0, False: 0]
Branch (223:20): [True: 0, False: 0]
Branch (223:20): [True: 0, False: 0]
Branch (223:20): [True: 0, False: 0]
Branch (223:20): [True: 0, False: 0]
|
224 | 2 | let msg = messages.remove(i); |
225 | 2 | union_messages.insert(pos + i, msg); |
226 | 33 | } |
227 | | } |
228 | 28 | trace!(">> Compiling messages with no references to unions..."); |
229 | 56 | for v33 in messages { |
230 | 33 | trace!({model=?&v}, "Compiling message"); |
231 | 33 | let v28 = Rc::new(Message::from_model(&proto, v)?5 ); |
232 | 28 | proto.messages.insert(v); |
233 | | } |
234 | 20 | } |
235 | 43 | if let Some(unions10 ) = value.unions { Branch (235:16): [True: 3, False: 17]
Branch (235:16): [Folded - Ignored]
Branch (235:16): [True: 3, False: 15]
Branch (235:16): [Folded - Ignored]
Branch (235:16): [True: 0, False: 0]
Branch (235:16): [True: 0, False: 0]
Branch (235:16): [True: 0, False: 0]
Branch (235:16): [True: 0, False: 0]
Branch (235:16): [True: 0, False: 0]
Branch (235:16): [True: 4, False: 1]
Branch (235:16): [True: 0, False: 0]
|
236 | 10 | trace!(">> Compiling unions..."); |
237 | 20 | for v10 in unions { |
238 | 10 | trace!({model=?&v}, "Compiling union"); |
239 | 10 | let v = Rc::new(Union::from_model(&proto, v)?0 ); |
240 | 10 | proto.unions.insert(v); |
241 | | } |
242 | 33 | } |
243 | 43 | trace!(">> Compiling messages with references to unions..."); |
244 | 49 | for (_, msg11 ) in union_messages { |
245 | 11 | trace!({model=?&msg}, "Compiling message"); |
246 | 11 | let v6 = Rc::new(Message::from_model(&proto, msg)?5 ); |
247 | 6 | proto.messages.insert(v); |
248 | | } |
249 | | |
250 | 38 | info!("Running list sanitizer pass..."); |
251 | 38 | for msg33 in proto.messages.iter() { |
252 | 33 | if msg.is_embedded() { Branch (252:16): [True: 4, False: 14]
Branch (252:16): [Folded - Ignored]
Branch (252:16): [True: 4, False: 11]
Branch (252:16): [Folded - Ignored]
Branch (252:16): [True: 0, False: 0]
Branch (252:16): [True: 0, False: 0]
Branch (252:16): [True: 0, False: 0]
Branch (252:16): [True: 0, False: 0]
Branch (252:16): [True: 0, False: 0]
Branch (252:16): [True: 0, False: 0]
Branch (252:16): [True: 0, False: 0]
|
253 | 16 | for field in &msg.fields8 { |
254 | 16 | let flag = match &field.ty { |
255 | 2 | FieldType::Container(v) => v.nested, |
256 | 14 | _ => true, |
257 | | }; |
258 | 16 | if !flag { Branch (258:24): [True: 0, False: 8]
Branch (258:24): [Folded - Ignored]
Branch (258:24): [True: 0, False: 8]
Branch (258:24): [Folded - Ignored]
Branch (258:24): [True: 0, False: 0]
Branch (258:24): [True: 0, False: 0]
Branch (258:24): [True: 0, False: 0]
Branch (258:24): [True: 0, False: 0]
Branch (258:24): [True: 0, False: 0]
Branch (258:24): [True: 0, False: 0]
Branch (258:24): [True: 0, False: 0]
|
259 | 0 | return Err(Error::MissingNestedList(format!("{}::{}", msg.name, field.name))); |
260 | 16 | } |
261 | | } |
262 | 25 | } |
263 | | } |
264 | 38 | Ok(proto) |
265 | 58 | } |
266 | | } |